P1: ALU

Add Comment

#### 8/29/2022

#### ∨ Details

**Reminder:** This is an individual project. Do not show your work to anyone or look at the work of anyone (in the class or online). Everything you submit must be implemented by you personally.



## Overview

In this project, you'll design a RISC-V ALU. **RISC-V** is an instruction set architecture (ISA) that defines the functions a computer can carry out through assembly instructions. ALAL Color in the card begin unit, legicles many of the Arra computations dictated by those instructions. So in short, you'll be building a key component of a CPU. Throughout this project and the next, you'll move from small, special-purpose circuits to the full, complex CPU in all its glory!

The program you will use to create your Attain opising, a free hardware design and circuit simulation tool. If you haven't yet, you can download it from the <u>Electronic Resources (https://canva.comeli.edu/courses/42905/pages/electronic-resources)</u>. Logisim comes with libraries containing basic gates, memory chips, multiplexers and decoders, and other simple components.

# However, for this assignment you may ON is set the following Logisim elements: CS

- Anything in the **Wiring** folder *except*: constants, and anything smaller/less abstract than a gate including but not limited to the resistor, power, ground and transistor elements.
- Anything in the Base folder (wires, text, etc.)
- · Anything in the Gates folder except: the even parity, and parity, and controlled buffer elements
- · Anything in the Plexers folder

**Important** - Use this guideline (https://canvas.cornell.edu/courses/42905/pages/logisim-design-guidelines) for logisim FAQs and best practices for circuit design to avoid losing points!

## What to Submit

All of the following should be subcircuits in a single Logisim circuit file. To get you started we've provided an <a href="ALU template.circ">ALU template.circ</a> (<a href="https://canvas.cornell.edu/courses/42905/files/6387723/download?download\_frd=1">https://canvas.cornell.edu/courses/42905/files/6387723/download?download\_frd=1</a>) file with all the appropriate inputs and outputs specified.

- A single Logisim project file containing your ALU and all needed subcomponents. Please ensure that your circuit has no external dependencies!
  - We will use Logisim's test vector function to test your circuits. In order for it to work correctly, you must ensure that the Logisim circuits containing your solutions are named **precisely** "LeftShift32", "Shift32", "Add32", and "ALU32", and the inputs/outputs are named "A", "B", "Op", "Sa", "C", and "V".
- · A design document that details the implementation of your circuit.

| < |  | > |
|---|--|---|
|   |  |   |

• Heads up! Last semester, students who filled out the survey reported the following:

How long did you (personally) spend working on this project?



## Circuit Design

## Circuit 1: Add32

| Add32:   | C = A + B + Cin; $V = overflow$ |
|----------|---------------------------------|
| Inputs:  | A[32], B[32], Cin               |
| Outputs: | C[32], V                        |

The output C is computed by adding A, B, and Cin. A, B, and C are **signed two's complement numbers.** If overflow occurs, the output V should be asserted. In such cases, the output C should correspond to the value computed if all overflow errors are ignored. Once you complete your adder, be sure to test it!

Hint: Use subcomponents to make wiring easier by building a 1-bit adder, then a 2-bit adder, then a 4-bit adder, and so on up to 32-bits.

Signed vs. Unsigned Adders
Assignment Project Exam Help

There actually isn't a huge difference between signed and unsigned adders. In fact the only difference between the two is the way that overflow is calculated. Overflow should always be calculated using the two most significant bits of a number. When building to a 32-bit adder, make sure to keep this in mind and do not bury your overflow bits within your adder. Below is a 4-bit unsigned adder, like the one you did in Lab 1...



...and here is a 4-bit two's complement signed adder.



>

Use this distinction when building your 32-bit signed adder. Note that the one-bit adder subcircuits in both the signed and unsigned 4-bit adders are identical (the one-bit adders are all **unsigned** adders).

## Circuit 2: Shift32 (Logical)

| Inputs:  | B[32], Sa[5], Cin, SL                                                                                 |
|----------|-------------------------------------------------------------------------------------------------------|
| Outputs: | C[32]                                                                                                 |
| Shift32: | C = (SL == 0) ? (B << Sa)   carrybits : carrybits   (B >>> Sa) Explanation: if $SL == 0$ , shift left |
|          | if SI == 1 shift right                                                                                |

You might remember during <u>Lab 2 (https://canvas.cornell.edu/courses/42905/assignments/381697)</u> implementing a LeftShift32. If you haven't already done so, we would recommend you merge over your LeftShift32 circuit to your ALU project as although we will be describing the Shift variant below, you will need to submit a finished alu.circ file with a subcircuit labeled LeftShift32.

The output C is computed by shifting B to the left Sa bits if SL == 0, and filling the vacated bits on the right with Carrybits, which is just Sa copies of Cin. Otherwise, the output C is computed by shifting B to the right Sa bits if SL == 1, and filling the vacated bits on the left with Carrybits. The shift amount Sa can be anything from 0 to 31, encoded as an **unsigned** integer.

When the specification denotes a pin as [B[32]], the pin should be named "B" and be 32 bits wide.

Once you are confident in your circuit's correctness, think about now how you would modify this circuit in order to handle both right shift arithmetic in addition to right shift logical and left shift. Do **not** create two different circuits to handle them.

Note: You should implement this shift Behalid Induction the logical and a intention variants & include shift) using Louis ready made LeftShift circuit. The way to do so is a challenge we leave up to you to figure out. You will be penalized if you implement Shift32 without using LeftShift32.

Hint: Shifting a value by a constant amount, either left or right, is simply a matter of adding, removing, and renaming the wires, and so requires no gates at all (this can be done with a specific simple s

## Circuit 3: ALU32

ALU32: CStutorcs
Inputs: A[32], B[32], Op[4], Sa[5]
Outputs: C[32], V

The c and v outputs are computed according to the value of the op input based on the table of operations below. For the add and subtract operations, the v output should be set to 1 if and only if the c output could be incorrect due to a numerical overflow occurring during computation. The value c output in the presence of overflow should correspond to the value computed if all overflow errors are ignored.

| Ор   | name                   | С                                       | meaning                                                                | V     |
|------|------------------------|-----------------------------------------|------------------------------------------------------------------------|-------|
| 0101 | and                    | C = A & B                               | C gets the value of the <b>bitwise</b> and operation of inputs A and B | V = 0 |
| 0100 | or                     | C = A   B                               |                                                                        | V = 0 |
| 100x | shift left logical     | C = B << Sa                             |                                                                        | V = 0 |
| 0010 | xor                    | C = A ^ B                               |                                                                        | V = 0 |
| 0011 | nor                    | C = ~(A   B)                            |                                                                        | V = 0 |
| 1100 | shift right logical    | C = B >>> Sa                            |                                                                        | V = 0 |
| 1101 | shift right arithmetic | C = B >> Sa                             |                                                                        | V = 0 |
| 0000 | ne                     | C = (A != B) ? 0000001 : 0000000        | if A!= B, C gets the value 0000001, else 0000000                       | V = 0 |
| 0001 | eq                     | C = (A == B) ? 0000001 : 0000000        |                                                                        | V = 0 |
| 1110 | le                     | C = (A ≤ 0) ? 0000001 : 0000000         |                                                                        | V = 0 |
| 1111 | at                     | $C = (A > 0) 2.000 0001 \cdot 000 0000$ |                                                                        | V = 0 |

An x in the opcode indicates that the circuit should not depend on the value of that bit when determining the appropriate operation. For example, your circuit should add when the opcode is either 0111 or 0110. All opcodes that are not covered in the table can have undefined behavior and will not be tested.

The expression (test)? 1:0 has a value of 1 if test is true, and 0 otherwise. In both cases, the upper 31 bits of the output are zero. Note the difference between logical right shift (which fills with zero bits), and arithmetic right shift (which fills the new bits with copies of the sign bit of B). The logical and (8), or (1), xor (^), nor, and complement (~) operators are all bit-wise operations. Also note that le and gt compare A to 0, not B.

## Notes & Hints

Getting started: Design your circuits on paper before building them in Logisim. Design, build, and test an adder/subtractor unit for use in your ALU. Repeat the same steps for circuit 2: design, then build and test the left shifter circuit first. Next, design, build, and test a left/right shifting unit to be used in the ALU. Think of the left/right shifter as miniature ALUs: it will have its own opcode-like control input of your choice that selects between its different sub-operations. Then design a comparator unit that can perform the four comparison operations by processing the output of the adder/subtractor or other subcomponents. Finally, design, build, and test the complete ALU for circuit 3. The overall idea is to compute several potentially needed values of the output C using the pieces you have already built and then to select the appropriate one using a multiplexer.

**Decoding logic:** Once you have all of your subcomponents, you need to combine these so that your circuit can compute several values in parallel, but it will ultimately select only one for output. Your decoding logic can often be simplified if you note that you only need the output of a subcomponent to be correct (i.e. for it to receive the correct inputs) if you know ultimately that it will be selected for output. In short, try to find the cases where you really don't care about the inputs to, or outputs from, a sub-component.

Don't duplicate components: Your ALU should use your adder and shifter as components. Your ALU should also use a single component of each for the entire design. As in class, your ALU should only use a single 32-bit adder component to implement both addition and subtraction. Similarly, your ALU should use only a single 32-bit shifter component to implement all of the shift operations. Do not change your original shifting and adding components, instead your analysis of the set of the shifter. You will be penalized if your final ALU circuit uses more than one adder or shifter.

On specifications: It is important that your implementation of the three circuits described above adhere to the specification in this document. Adhering to specification is important mimost all design processes, and it will also have a more immediate effect on your grade for this project. Automated grading will expect that the three circuits above (and their inputs and outputs) are named exactly as specified (case-sensitive!) and behave exactly as specified. Also recall that when the specification denotes a pin as A[32], the pin should be named "A" and be 32 bits wide. The pin should not be named "A[32]".

On circuit design: Be sure to follow the <u>logisim design guide (https://canvas.cornell.edu/courses/42905/pages/logisim-design-guidelines)</u> to avoid losing points!

On bringing components together: Check out the textbook to understand how an ALU works.

## **Test Vectors**

Extensively testing your circuit is important to ensure correctness.

While you obviously can't test every possible input, it is feasible in Logisim to test up to several thousand input tuples. You can even write a program to do so (in Python, Java, Bash, etc.) if you would like to, although we don't require it. You should strive to choose inputs strategically and include enough of them to test the *entire* functionality of your ALU. Some of your tuples should be written by hand to test corner cases (e.g. adding combinations of zero, +1, -1, max\_int, min\_int, etc.). Some might be generated systematically (e.g. testing every possible shift amount, and every possible op), and others might be generated randomly (to catch cases you didn't think of). **Testing is an important part of this project and you will be graded on both your random and edge cases.** 

For this project, you should create three ASCII text test vector files, one for each of the LeftShifter, Adder, and ALU circuits. A brief comment at the beginning of each file should indicate how the tests were chosen/generated and what they are testing. In addition, please clearly group your random and edge cases together and comment where each type begins (e.g. "#Random cases start here"). You can create a comment by starting a line with a pound sign (#).

The box below demonstrates the format of a Logisim test vector.

| (https://gapyas.compil.edu/courses/42005/modules/items/1511700) | (https://capyas.compoll.edu/courses/42005/modules/items/1 | 1511702) |
|-----------------------------------------------------------------|-----------------------------------------------------------|----------|
| <                                                               | >                                                         |          |
|                                                                 |                                                           |          |
|                                                                 |                                                           |          |

#if length of input value == width of input/output pin: binary #else: decimal #Choose whichever base(s) is/are most comfortable for you. #They will all be interpreted in binary by Logisim and you can assume this #will always work. The purpose of this is that, for instance, binary might be #easier for you if you are making tests for OR whereas decimal might be easier #if you're making tests for ADD. We assume that, unless you are Prof. Bracy, #your brain defaults to 7+6=13, not 0b111+0b110=0b1101, for instance. #In this example, the columns are aligned neatly for readability. #Your test vector does NOT have to be aligned like this. You can separate each #value with a single whitespace. We won't be grading your test on its looks. B[32] Cin C[32] Sa[5] #Decimal 34 0 136 325948595 15 -900104192 0 #Negative Decimal 00100 0000000000000000000000000000011111 #Binary 0x00000005 0x01 0x0 0x0000000A #Hexadecimal 0o1777777777 0<sub>0</sub>1 000 0o3777777776 #0ctal 000111110010101000101101111100010010 0x0 707653888 #Everything together!

#### Testing FAQ's

#### Q: How do we test our circuits?

The Cornell version of Logisim adds a special "Test Vector" feature to Logisim for automated testing of circuits. The documentation for this is accessed from within Logisim: select Help->User's Guide from the toolbar. On the left pane of the help window that appears, look for and select the item labeled "Test Vectors".

Q: To write a test case for my test vector, I need to know what the correct result for a certain operation is. How could I ever do this?

All of the ALU's operations are clearly defined arithmetic operations. The results can easily be computed by hand. Even better, implementations of these arithmetic operations are available in every major programming language.

Q: We can just use Logisim's logging feature to generate a test vector, right?

Let's get this straight. To verify the correctness of *your ALU*, you are going to log the output of *your ALU* for a few inputs, and then you are going to verify that *your ALU* cases the same outputs as your ALU?"

The first rule of tautology club is the first rule of tautology club (http://xkcd.com/703/).

Q: How many test cases do we need alta manber of tests enought CS

We aren't looking for a specific number of tests, but that your tests should convince you your circuit behaves as intended. That means you need to write both random and edge cases for each op code. The number of edge cases should vary depending on the op code but try to think of at least 3-4 cases that can cause unexpected behaviour (eg. overflow). Roughly 100 random cases split evenly between the op codes is definitely enough, but you are welcome to generate more for parrown testing curposes.

## **Design Documentation**

Your final submission will include a design doc explaining the following:

- Diagrams showing the components of your ALU (adder, shifter etc.) and connections between them. Explain any functionality that we might find confusing.
  - Note that this, and all diagrams, can simply be Logisim screenshots if your circuit is neat and easy to read. Please use the Logisim text tool or an image editor to add labels/annotations if necessary.
  - If a diagram contains many components, a couple of sentences should suffice for the description of its functionality. You definitely do not need to write paragraphs explaining how everything works.
- Explanations of your control logic. What exactly does this mux do? What effect does changing this bit have? Include a brief description of what the values of any control signal mean, and a truth table showing the value that signal takes for each possible opcode.
- Any design decisions or tradeoffs you made, if you feel they are relevant.

Note that the design doc does *not* have to be long or overly detailed, nor does it have to be in LaTeX. Here is an example format <a href="https://canvas.cornell.edu/courses/42905/files/6057292/download?wrap=1">https://canvas.cornell.edu/courses/42905/files/6057292/download?wrap=1</a> \(\frac{\text{https://canvas.cornell.edu/courses/42905/files/6057601/download?wrap=1}\) \(\frac{\text{https://canvas.cornell.edu/courses/42905/files/6057601/download?wrap=1}\)\) \(\text{https://canvas.cornell.edu/courses/42905/files/6057601/download?download?rd=1}\)\)\)\)

Nou don't have to follow it.

Students often ask how detailed a design doc should be. To give you an idea, good design docs tend to be around 10 pages in length with full-sized screenshots. That being said, reaching this page number is not indicative of the quality of an individual design doc. Shorter ones can be good if filled with concise yet comprehensive descriptions and longer ones can be bad if missing key information.

| , | _ |  |
|---|---|--|
| ` |   |  |

In addition to a design document, you are required to submit a short README. The README includes the following:

- · Your name and NetID
- · An estimate of length of the critical path of the complete ALU
- · An estimate of the number of gates required to implement the ALU (including gates needed for subcomponents)

### Critical Path

In synchronous logic, the critical path is the slowest logic path in the circuit. We have assumed that the operation of the ALU completes in one clock cycle. In order to determine how long the clock cycle is, you need to figure out which path in your circuit is the longest path for the input signals to propagate through. This particular path is called the critical path. The amount of time for the input signals to propagate through the critical path is the minimum length of one clock cycle. The reciprocal of the clock period gives the maximum frequency of the input clock signal. You may express your critical path in terms of the number of gates in the path. To determine the critical path you should use the following simplifying assumptions:

- Standard AND, OR, NOR, NAND, XOR, and XNOR gates have a gate delay of one unit
- · NOT gates are ignored
- · Multi-input and multi-bit gates both have the same gate delay as their variants
- A mux has a gate delay of 2 regardless of size (you can derive this formula on your own by looking at how a mux is built out of basic gates!)

#### Gate Count

In microprocessor design, gate count refers to the number of transistor switches, or gates, that are needed to implement a design. Even with today's process technology, gate counts remain one of the most important overall factors in the end price of a chip. Designs with fewer gates will typically cost less, and for this reason, gate count remains a commonly used metric in the industry. To determine the gate count you should use the following assumptions:

- Standard AND, OR, NOR, NAND, XOR and XNOR gates count as one gate
  NOT gates are ignored ssignment Project Exam Help
- Multi-input gates count as a single gate
- · An n-bit gate counts as n gates (the multi-bit gates Logisim provides aren't actually real; they are just a convenient shorthand for using a gate
- A mux counts as (# of data bits)\*(# of the state of the

# Submission Checklist WeChat: cstutorcs

### Circuit

- Follows every item in <u>Logisim Design Guidelines (https://canvas.cornell.edu/courses/42905/pages/logisim-design-guidelines)</u>
- · Does not have any forbidden components (e.g. constants, Logisim Adder/Shifter)
- · Add32, LeftShift32, and Shift32 subcircuits should appear only once in the entire submission
- · Any operations that can use the same circuitry (e.g. EQ and NEQ, ADD and SUB) should use the same circuitry

Note: Any circuitry that mimics the function of a forbidden or duplicated component will still be penalized.

- · Does not use more components than necessary
- · No overcomplicated circuitry
- · No circuitry with the same functions as a Logisim component (e.g. building your own MUX)
- · Add32, LeftShift32, Shift32, and ALU32 subcircuits follow specifications exactly
- Should be named exactly as shown (case sensitive)
- Should have all the pins listed, and only the pins listed, also named correctly (case sensitive, without bit widths e.g. "[32]")
- Is readable, using abstractions (subcircuits) appropriately and tunnels sparingly

#### Documentation

- · Includes README with gate count and critical path
- · Design document adequately describes subcircuits and the logic used to control them

#### Testing

- Includes comments that explain how tests were chosen
- Edge cases include explanations on why they are edge cases

| < |  | > |
|---|--|---|
|   |  |   |

Academic Integrity. As one of the most widely studied architectures, RISC-V has a wealth of information available on the web and in textbooks. You may consult any RISC-V documentation available to you in order to learn about the instruction set, what each instruction does, how an ALU works, etc. However, we expect your design to be entirely your own, and your submission should cite any significant sources of information you used. If you are unsure if it is okay to borrow from some other source, just ask the TAs. If you are hesitant to ask the TAs or to cite the source, then it is probably not okay. Plagiarism in any form will not be tolerated. It is also your responsibility to make sure your sources match the material we describe here (warning: the top Google hit for "RISC reference" contains several inaccuracies).

(https://canvas.cornell.edu/courses/42905/assignments/381719)

P2 (https://canvas.cornell.edu/courses/42905/assignments/381718)

(https://canvas.cornell.edu/courses/42905/assignments/381719)

# Assignment Project Exam Help

https://tutorcs.com

WeChat: cstutorcs

